/*
 * To change this template, choose Tools | Templates
 * and open the template in the editor.
 */
package stud.cp.user.dbaccess;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.LinkedList;
import java.util.List;
import org.apache.log4j.Logger;
import stud.cp.core.dbaccess.DAO;
import stud.cp.core.dbaccess.DAOException;
import stud.cp.user.valueobjects.Weapon;

/**
 * Класс, описывающий доступ к таблице Weapon
 * @author Morfant
 */
public class WeaponDAO implements DAO<Weapon> {

    /**
     * Логгер.
     */
    private static Logger logger = Logger.getLogger("CPTNP");
    /**
     * Имя колонок таблицы weapon
     */
    private static String[] weaponColumnName = {"weapon_id", "name", "description", "weapon_type_id"};
    /**
     * Список названий колонок таблицы weapon_types.
     */
    private static String[] weaponTypesColumnName = {"weapon_type_id", "weapon_type_name", "description"};
    /**
     * Коннект к базе данных.
     */
    private Connection connection = null;

    /**
     * Конструктор, который устанавливает
     * коннект к базе данных.
     * @param connection Связь с базой данных.
     */
    public WeaponDAO(Connection connection) {
        this.connection = connection;
    }

    /**
     * Запрос, позволяющий получить запись по id.
     * @param id ID
     * @return Запрос.
     */
    private String getQueryForGetById(int id) {
        StringBuilder query = new StringBuilder();

        query.append("select w.weapon_id,w.name,w.description,w.weapon_type_id,wt.name as \"weapon_type_name\"\n");
        query.append("from weapons w \n");
        query.append("inner join weapon_types wt on w.weapon_type_id=wt.weapon_type_id \n");
        query.append("where weapon_id=");
        query.append(id);

        return query.toString();
    }

    /**
     * Метод, позволяющий получить запись
     * по её ID
     * @param id ID
     * @return Данные записи.
     * @throws DAOException
     * <p>
     *  Ошикба доступа к данным.
     * </p>
     */
    @Override
    public Weapon getById(int id) throws DAOException {
        Weapon weapon = null;
        ResultSet set = null;
        PreparedStatement prestat = null;
        try {
            prestat = connection.prepareStatement(getQueryForGetById(id));
            set = prestat.executeQuery();
            if (set.next()) {
                weapon = new Weapon(set.getInt(weaponColumnName[0]),
                        set.getInt(weaponColumnName[3]),
                        set.getString(weaponColumnName[1]),
                        set.getString(weaponColumnName[2]),
                        set.getString(weaponTypesColumnName[1]));
            }
        } catch (SQLException ex) {
            logger.debug("DAO_propety", ex);
            throw new DAOException("Ошибка при создание данных..", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
        return weapon;
    }

    /**
     * Запрос на обновление записи в таблице.
     * @param id ID
     * @param name Имя
     * @return Запрос.
     */
    private String getQueryForUpdate(Integer weapon_id, String name, String description) {
        StringBuilder query = new StringBuilder();
        query.append("UPDATE weapons \n");
        query.append("SET name='");
        query.append(name);
        query.append("',");
        query.append("description='");
        query.append(description);
        query.append("'\n");
        query.append("WHERE weapon_id=");
        query.append(weapon_id);
        return query.toString();
    }

    /**
     * Метод, отвечающий за обновление данных.
     * @param entry Запись.
     * @throws DAOException Ошибка при обновление данных.
     */
    @Override
    public void update(Weapon entry) throws DAOException {
        PreparedStatement prestat = null;
        try {
            prestat = connection.prepareStatement(
                    getQueryForUpdate(
                    entry.getWeapon_id(),
                    entry.getName(),
                    entry.getDescription()));
            prestat.executeUpdate();
        } catch (SQLException ex) {
            logger.debug("DAO_propety_group", ex);
            throw new DAOException("Ошибка обновлени данных.", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
    }

    /**
     * Запрос на добавление
     * новой записи в таблицу.
     * @param id ID
     * @param name Имя
     * @return Запрос.
     */
    private String getQueryForInsert(Integer weapon_id, Integer weapon_type_id, String name, String description) {
        StringBuilder query = new StringBuilder();
        query.append("insert into weapons(weapon_id,weapon_type_id,name,description) \n");
        query.append("VALUES(");
        query.append(weapon_id);
        query.append(",");
        query.append(weapon_type_id);
        query.append(",'");
        query.append(name);
        query.append("','");
        query.append(description);
        query.append("')");
        return query.toString();
    }

    /**
     * Зарпос на получения следующего ID.
     * @return Зарпос.
     */
    private String getQueryForSelectNextID() {
        StringBuilder query = new StringBuilder();
        query.append("SELECT weapons_seq.NEXTVAL from dual \n");
        return query.toString();
    }

    /**
     * Зарпос на получения максимального id из таблицы.
     * @return Зарпос.
     */
    private String getQueryForSelectMaxId() {
        StringBuilder query = new StringBuilder();
        query.append("SELECT MAX(weapon_id) \n");
        query.append("FROM weapons");
        return query.toString();
    }

    /**
     * Метод, который создает новую запись в таблице.
     * @param entry Запись.
     * @throws DAOException Ошибка создания новой записи.
     */
    @Override
    public int create(Weapon entry) throws DAOException {
        ResultSet set = null;
        PreparedStatement prestat = null;
        try {
            prestat = connection.prepareStatement(getQueryForSelectNextID());
            set = prestat.executeQuery();
            Integer index = null;
            if (set.next()) {
                index = set.getInt(1);
            } else {
                index = 0;
            }
            prestat.close();
            prestat = connection.prepareStatement(getQueryForInsert(index,
                    entry.getWeapon_type_id(),
                    entry.getName(),
                    entry.getDescription()));
            prestat.executeUpdate();
            //TODO::[High]Delete string
            entry.setWeaon_id(index);
            return index;
        } catch (SQLException ex) {
            logger.debug("DAO_propety_group", ex);
            throw new DAOException("Ошибка при создание записи.", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
    }

    @Override
    public List<Weapon> getAll() throws DAOException {
        throw new UnsupportedOperationException("Not supported yet.");
    }

    /**
     * Запрос, позволяющий при выполнении
     * получить все оружие, которое принадлежит определенному типу оружия.
     * @param weapon_type_id ID типа оружия
     * @return Строка запроса.
     */
    private String getQueryForAllWeaponForWeaponType(int weapon_type_id) {
        StringBuilder query = new StringBuilder();
        query.append("select w.weapon_id,w.name,w.description,w.weapon_type_id,wt.name as \"weapon_type_name\"\n");
        query.append("from weapons w \n");
        query.append("inner join weapon_types wt on w.weapon_type_id=wt.weapon_type_id \n");
        query.append("where wt.weapon_type_id=");
        query.append(weapon_type_id);
        return query.toString();
    }

    /**
     * Метод, который возращает список
     * оружия принадлежащий определенному типу оружия.
     * @param weapon_type_id ID типа оружия.
     * @return Список данных.
     */
    public List<Weapon> getAllWeaponByWeaponType(int weapon_type_id) throws DAOException {
        List<Weapon> data_list = new LinkedList<Weapon>();
        ResultSet set = null;
        PreparedStatement prestat = null;
        try {
            prestat = connection.prepareStatement(getQueryForAllWeaponForWeaponType(weapon_type_id));
            set = prestat.executeQuery();
            while (set.next()) {
                data_list.add(new Weapon(set.getInt(weaponColumnName[0]),
                        set.getInt(weaponColumnName[3]),
                        set.getString(weaponColumnName[1]),
                        set.getString(weaponColumnName[2]),
                        set.getString(weaponTypesColumnName[1])));
            }
        } catch (SQLException ex) {
            logger.debug("DAO_propeties", ex);
            throw new DAOException("Ошибка получения списка данных.", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
        return data_list;
    }

    /**
     * Запрос, позволяющий при выполнении
     * получить все оружие, которое принадлежит определенному типу оружия.
     * @param weapon_type_id ID типа оружия
     * @return Строка запроса.
     */
    private String getQueryForAllWeaponForWeaponTypeExcludeCurrentWeapon(int weapon_type_id, int weapon_id) {
        StringBuilder query = new StringBuilder();
        query.append("select w.weapon_id,w.name,w.description,w.weapon_type_id,wt.name as \"weapon_type_name\"\n");
        query.append("from weapons w \n");
        query.append("inner join weapon_types wt on w.weapon_type_id=wt.weapon_type_id \n");
        query.append("where wt.weapon_type_id=");
        query.append(weapon_type_id);
        query.append("and w.weapon_id<>");
        query.append(weapon_id);
        return query.toString();
    }

    /**
     * Метод, который возращает список
     * оружия принадлежащий определенному типу оружия,
     * исключая из него оружие, которое запрашивается пользователем.
     * @param weapon_type_id ID типа оружия.
     * @param weapon_id ID оружия, которое запрашивает список ссылок.
     * @return Список данных.
     */
    public List<Weapon> getAllWeaponByWeaponTypeExcludeWeapon(int weapon_type_id, int weapon_id) throws DAOException {
        List<Weapon> data_list = new LinkedList<Weapon>();
        ResultSet set = null;
        PreparedStatement prestat = null;
        try {
            prestat =
                    connection.prepareStatement(
                    getQueryForAllWeaponForWeaponTypeExcludeCurrentWeapon(weapon_type_id, weapon_id));
            set = prestat.executeQuery();
            while (set.next()) {
                data_list.add(new Weapon(set.getInt(weaponColumnName[0]),
                        set.getInt(weaponColumnName[3]),
                        set.getString(weaponColumnName[1]),
                        set.getString(weaponColumnName[2]),
                        set.getString(weaponTypesColumnName[1])));
            }
        } catch (SQLException ex) {
            logger.debug("DAO_propeties", ex);
            throw new DAOException("Ошибка получения списка данных.", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
        return data_list;
    }

    /**
     * Зарпос на удаление из таблицы
     * записи по id.
     * @param id ID
     * @return Зарпос.
     */
    private String getQueryForDelete(int id) {
        StringBuilder query = new StringBuilder();
        query.append("DELETE FROM weapons \n");
        query.append("WHERE weapon_id=");
        query.append(id);
        return query.toString();
    }

    /**
     * Удаление из табилцы
     * property_values.
     * @param weapon_id ID оружия.
     * @return  Зарпос.
     */
    private String getQueryForDeleteFromPropertyValues(int weapon_id) {
        StringBuilder query = new StringBuilder();
        query.append("DELETE FROM property_values \n");
        query.append("WHERE weapon_id=");
        query.append(weapon_id);
        return query.toString();
    }

    /**
     * Удаление из табилцы
     * weapon_references.
     * @param weapon_id ID оружия.
     * @return  Зарпос.
     */
    private String getQueryForDeleteFromWeaponReferences(int weapon_id) {
        StringBuilder query = new StringBuilder();
        query.append("DELETE FROM weapon_references \n");
        query.append("WHERE weapon_id=");
        query.append(weapon_id);
        query.append("\n OR reference_weapon_id=");
        query.append(weapon_id);
        return query.toString();
    }

    /**
     * Удаление истории оружия.
     * @param id ID оружия.
     * @return Запрос.
     */
    private String getQueryForDeleteWeaponHistory(int id) {
        StringBuilder query = new StringBuilder();
        query.append("DELETE FROM history \n");
        query.append("WHERE weapon_id=");
        query.append(id);
        return query.toString();
    }

    /**
     * Операция удаления записи из
     * таблицы.
     * @param id Объект, который надо удалить.
     */
    @Override
    public void delete(int id) throws DAOException {
        PreparedStatement prestat = null;
        try {
            prestat = connection.prepareStatement(getQueryForDeleteWeaponHistory(id));
            prestat.executeUpdate();
            prestat.close();
            prestat = connection.prepareStatement(getQueryForDeleteFromPropertyValues(id));
            prestat.executeUpdate();
            prestat.close();
            prestat = connection.prepareStatement(getQueryForDeleteFromWeaponReferences(id));
            prestat.executeUpdate();
            prestat.close();
            prestat = connection.prepareStatement(getQueryForDelete(id));
            prestat.executeUpdate();
        } catch (SQLException ex) {
            logger.debug("DAO_propety_group", ex);
            throw new DAOException("Ошибка удаления данных.", ex);
        } finally {
            if (prestat != null) {
                try {
                    prestat.close();
                } catch (SQLException ex) {
                    logger.debug("DAO_propety", ex);
                    throw new DAOException("Ошибка закрытия ResultSet`а.", ex);
                }
            }
        }
    }
}
